/*
 * Copyright (c) 2015, 2024, Oracle and/or its affiliates.
 *
 * This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License, version 2.0, as published by
 * the Free Software Foundation.
 *
 * This program is designed to work with certain software that is licensed under separate terms, as designated in a particular file or component or in
 * included license documentation. The authors of MySQL hereby grant you an additional permission to link the program and your derivative works with the
 * separately licensed software that they have either included with the program or referenced in the documentation.
 *
 * Without limiting anything contained in the foregoing, this file, which is part of MySQL Connector/J, is also subject to the Universal FOSS Exception,
 * version 1.0, a copy of which can be found at http://oss.oracle.com/licenses/universal-foss-exception.
 *
 * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License, version 2.0, for more details.
 *
 * You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc.,
 * 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
 */

package testsuite.x.internal;

import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertNotNull;
import static org.junit.jupiter.api.Assertions.assertNull;
import static org.junit.jupiter.api.Assumptions.assumeTrue;

import java.io.IOException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.function.Consumer;
import java.util.function.Supplier;

import org.junit.jupiter.api.AfterEach;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Tag;
import org.junit.jupiter.api.Test;

import com.mysql.cj.conf.PropertyDefinitions;
import com.mysql.cj.protocol.ColumnDefinition;
import com.mysql.cj.protocol.ProtocolEntity;
import com.mysql.cj.protocol.ResultBuilder;
import com.mysql.cj.protocol.x.StatementExecuteOk;
import com.mysql.cj.protocol.x.StatementExecuteOkBuilder;
import com.mysql.cj.protocol.x.XMessageBuilder;
import com.mysql.cj.protocol.x.XProtocol;
import com.mysql.cj.result.DefaultColumnDefinition;
import com.mysql.cj.result.Field;
import com.mysql.cj.result.Row;
import com.mysql.cj.xdevapi.DocFilterParams;
import com.mysql.cj.xdevapi.RowResult;

/**
 * Tests for protocol-level <b>async</b> APIs against X Plugin via X Protocol.
 */
@Tag("Async")
public class XProtocolAsyncTest extends InternalXBaseTestCase {

    private XProtocol protocol;
    private XMessageBuilder messageBuilder;

    @BeforeEach
    public void setupTestProtocol() {
        if (this.isSetForXTests) {
            this.protocol = createAuthenticatedTestProtocol(createTestProtocol(), this.testProperties);
            this.messageBuilder = (XMessageBuilder) this.protocol.getMessageBuilder();
        }
    }

    @AfterEach
    public void destroyTestProtocol() throws IOException {
        if (this.isSetForXTests) {
            this.protocol.close();
        }
    }

    /**
     * Helper class to hold values across threads and closures.
     *
     * @param <T>
     */
    public static class ValueHolder<T> implements Consumer<T>, Supplier<T> {

        T val;

        @Override
        public void accept(T t) {
            this.val = t;
        }

        @Override
        public T get() {
            return this.val;
        }

    }

    @Test
    public void simpleSuccessfulQuery() throws Exception {
        assumeTrue(this.isSetForXTests, PropertyDefinitions.SYSP_testsuite_url_mysqlx + " must be set to run this test.");

        try {
            String collName = createTempTestCollection(this.protocol);

            String json = "{'_id': '85983efc2a9a11e5b345feff819cdc9f', 'testVal': 1, 'insertedBy': 'Jess'}".replaceAll("'", "\"");
            this.protocol.send(this.messageBuilder.buildDocInsert(getTestDatabase(), collName, Arrays.asList(new String[] { json }), false), 0);
            this.protocol.readQueryResult(new StatementExecuteOkBuilder());

            final ValueHolder<ColumnDefinition> metadataHolder = new ValueHolder<>();
            final ValueHolder<ArrayList<Row>> rowHolder = new ValueHolder<>();
            rowHolder.accept(new ArrayList<>());
            final ValueHolder<StatementExecuteOk> okHolder = new ValueHolder<>();
            final ValueHolder<Throwable> excHolder = new ValueHolder<>();

            this.protocol.queryAsync(this.messageBuilder.buildFind(new DocFilterParams(getTestDatabase(), collName)), new ResultBuilder<RowResult>() {

                private ArrayList<Field> fields = new ArrayList<>();
                private ColumnDefinition metadata;

                @Override
                public boolean addProtocolEntity(ProtocolEntity entity) {
                    if (entity instanceof Field) {
                        this.fields.add((Field) entity);

                    } else if (entity instanceof ColumnDefinition) {
                        this.metadata = (ColumnDefinition) entity;
                        metadataHolder.accept(this.metadata);

                    } else if (entity instanceof Row) {
                        if (this.metadata == null) {
                            this.metadata = new DefaultColumnDefinition(this.fields.toArray(new Field[] {}));
                            metadataHolder.accept(this.metadata);
                        }
                        rowHolder.get().add((Row) entity);

                    } else if (entity instanceof StatementExecuteOk) {
                        okHolder.accept((StatementExecuteOk) entity);
                        synchronized (XProtocolAsyncTest.this) {
                            XProtocolAsyncTest.this.notify();
                        }
                        return true;
                    }
                    return false;
                }

                @Override
                public RowResult build() {
                    return null;
                }

            });

            synchronized (this) {
                // timeout in case we get stuck
                this.wait(5000);
            }

            assertEquals(1, metadataHolder.get().getFields().length);
            assertEquals(1, rowHolder.get().size());
            assertNotNull(okHolder.get());
            assertNull(excHolder.get());
        } finally {
            dropTempTestCollection(this.protocol);
        }
    }

}
